1   /*
2    * Copyright 2002-2019 the original author or authors.
3    *
4    * Licensed under the Apache License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    *      https://www.apache.org/licenses/LICENSE-2.0
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  
17  package org.springframework.integration.jpa.core;
18  
19  import java.util.List;
20  
21  import javax.persistence.EntityManager;
22  import javax.persistence.EntityManagerFactory;
23  
24  import org.springframework.beans.BeansException;
25  import org.springframework.beans.factory.BeanFactory;
26  import org.springframework.beans.factory.BeanFactoryAware;
27  import org.springframework.beans.factory.InitializingBean;
28  import org.springframework.expression.EvaluationContext;
29  import org.springframework.expression.Expression;
30  import org.springframework.expression.common.LiteralExpression;
31  import org.springframework.integration.expression.ExpressionUtils;
32  import org.springframework.integration.jpa.support.JpaParameter;
33  import org.springframework.integration.jpa.support.PersistMode;
34  import org.springframework.integration.jpa.support.parametersource.BeanPropertyParameterSourceFactory;
35  import org.springframework.integration.jpa.support.parametersource.ExpressionEvaluatingParameterSourceFactory;
36  import org.springframework.integration.jpa.support.parametersource.ParameterSource;
37  import org.springframework.integration.jpa.support.parametersource.ParameterSourceFactory;
38  import org.springframework.lang.Nullable;
39  import org.springframework.messaging.Message;
40  import org.springframework.messaging.MessagingException;
41  import org.springframework.util.Assert;
42  import org.springframework.util.CollectionUtils;
43  
44  /**
45   * Executes Jpa Operations that produce payload objects from the result of the provided:
46   *
47   * <ul>
48   *     <li>entityClass</li>
49   *     <li>JpQl Select Query</li>
50   *     <li>Sql Native Query</li>
51   *     <li>JpQl Named Query</li>
52   *     <li>Sql Native Named Query</li>
53   * </ul>
54   *
55   * When objects are being retrieved, it also possibly to:
56   *
57   * <ul>
58   *     <li>delete the retrieved object</li>
59   * </ul>
60   *
61   * If neither entityClass nor any other query is specified then the entity-class
62   * is "guessed" from the {@link Message} payload.idExpression
63   *
64   * @author Gunnar Hillert
65   * @author Amol Nayak
66   * @author Artem Bilan
67   *
68   * @since 2.2
69   *
70   */
71  public class JpaExecutor implements InitializingBean, BeanFactoryAware {
72  
73  	private final JpaOperations jpaOperations;
74  
75  	private List<JpaParameter> jpaParameters;
76  
77  	private Class<?> entityClass;
78  
79  	private String jpaQuery;
80  
81  	private String nativeQuery;
82  
83  	private String namedQuery;
84  
85  	private Expression maxResultsExpression;
86  
87  	private Expression firstResultExpression;
88  
89  	private Expression idExpression;
90  
91  	private PersistMode persistMode = PersistMode.MERGE;
92  
93  	private ParameterSourceFactory parameterSourceFactory = null;
94  
95  	private ParameterSource parameterSource;
96  
97  	private boolean flush = false;
98  
99  	private int flushSize = 0;
100 
101 	private boolean clearOnFlush = false;
102 
103 	private boolean deleteAfterPoll = false;
104 
105 	private boolean deleteInBatch = false;
106 
107 	private boolean expectSingleResult = false;
108 
109 	/**
110 	 * Indicates that whether only the payload of the passed in {@link Message}
111 	 * will be used as a source of parameters. The is 'true' by default because as a
112 	 * default a {@link BeanPropertyParameterSourceFactory} implementation is
113 	 * used for the sqlParameterSourceFactory property.
114 	 */
115 	private Boolean usePayloadAsParameterSource = null;
116 
117 	private BeanFactory beanFactory;
118 
119 	private EvaluationContext evaluationContext;
120 
121 	/**
122 	 * Constructor taking an {@link EntityManagerFactory} from which the
123 	 * {@link EntityManager} can be obtained.
124 	 * @param entityManagerFactory Must not be null.
125 	 */
126 	public JpaExecutor(EntityManagerFactory entityManagerFactory) {
127 		Assert.notNull(entityManagerFactory, "entityManagerFactory must not be null.");
128 
129 		DefaultJpaOperations defaultJpaOperations = new DefaultJpaOperations();
130 		defaultJpaOperations.setEntityManagerFactory(entityManagerFactory);
131 		defaultJpaOperations.afterPropertiesSet();
132 
133 		this.jpaOperations = defaultJpaOperations;
134 	}
135 
136 	/**
137 	 * Constructor taking an {@link EntityManager} directly.
138 	 * @param entityManager Must not be null.
139 	 */
140 	public JpaExecutor(EntityManager entityManager) {
141 		Assert.notNull(entityManager, "entityManager must not be null.");
142 
143 		DefaultJpaOperations defaultJpaOperations = new DefaultJpaOperations();
144 		defaultJpaOperations.setEntityManager(entityManager);
145 		defaultJpaOperations.afterPropertiesSet();
146 		this.jpaOperations = defaultJpaOperations;
147 	}
148 
149 	/**
150 	 * If custom behavior is required a custom implementation of {@link JpaOperations}
151 	 * can be passed in. The implementations themselves typically provide access
152 	 * to the {@link EntityManager}.
153 	 * See also {@link DefaultJpaOperations} and {@link AbstractJpaOperations}.
154 	 * @param jpaOperations Must not be null.
155 	 */
156 	public JpaExecutor(JpaOperations jpaOperations) {
157 		Assert.notNull(jpaOperations, "jpaOperations must not be null.");
158 		this.jpaOperations = jpaOperations;
159 	}
160 
161 	public void setIntegrationEvaluationContext(EvaluationContext evaluationContext) {
162 		this.evaluationContext = evaluationContext;
163 	}
164 
165 	/**
166 	 * Sets the class type which is being used for retrieving entities from the
167 	 * database.
168 	 * @param entityClass Must not be null.
169 	 */
170 	public void setEntityClass(Class<?> entityClass) {
171 		Assert.notNull(entityClass, "entityClass must not be null.");
172 		this.entityClass = entityClass;
173 	}
174 
175 	/**
176 	 * @param jpaQuery The provided JPA query must neither be null nor empty.
177 	 */
178 	public void setJpaQuery(String jpaQuery) {
179 		Assert.isTrue(this.nativeQuery == null && this.namedQuery == null, "You can define only one of the "
180 				+ "properties 'jpaQuery', 'nativeQuery', 'namedQuery'");
181 		Assert.hasText(jpaQuery, "jpaQuery must neither be null nor empty.");
182 		this.jpaQuery = jpaQuery;
183 	}
184 
185 	/**
186 	 * You can also use native Sql queries to poll data from the database. If set
187 	 * this property will allow you to use native SQL. Optionally you can also set
188 	 * the entityClass property at the same time. If specified the entityClass will
189 	 * be used as the result class for the native query.
190 	 * @param nativeQuery The provided SQL query must neither be null nor empty.
191 	 */
192 	public void setNativeQuery(String nativeQuery) {
193 
194 		Assert.isTrue(this.namedQuery == null && this.jpaQuery == null, "You can define only one of the "
195 				+ "properties 'jpaQuery', 'nativeQuery', 'namedQuery'");
196 		Assert.hasText(nativeQuery, "nativeQuery must neither be null nor empty.");
197 
198 		this.nativeQuery = nativeQuery;
199 	}
200 
201 	/**
202 	 * A named query can either refer to a named JPQL based query or a native SQL
203 	 * query.
204 	 * @param namedQuery Must neither be null nor empty
205 	 */
206 	public void setNamedQuery(String namedQuery) {
207 
208 		Assert.isTrue(this.jpaQuery == null && this.nativeQuery == null, "You can define only one of the "
209 				+ "properties 'jpaQuery', 'nativeQuery', 'namedQuery'");
210 
211 		Assert.hasText(namedQuery, "namedQuery must neither be null nor empty.");
212 		this.namedQuery = namedQuery;
213 	}
214 
215 	public void setPersistMode(PersistMode persistMode) {
216 		this.persistMode = persistMode;
217 	}
218 
219 	public void setJpaParameters(List<JpaParameter> jpaParameters) {
220 		this.jpaParameters = jpaParameters;
221 	}
222 
223 	public void setUsePayloadAsParameterSource(Boolean usePayloadAsParameterSource) {
224 		this.usePayloadAsParameterSource = usePayloadAsParameterSource;
225 	}
226 
227 	/**
228 	 * If set to {@code true} the {@link javax.persistence.EntityManager#flush()} will be called
229 	 * after persistence operation.
230 	 * Has the same effect, if the {@link #flushSize} is specified to {@code 1}.
231 	 * For convenience in cases when the provided entity to persist is not an instance of {@link Iterable}.
232 	 * @param flush defaults to 'false'.
233 	 */
234 	public void setFlush(boolean flush) {
235 		this.flush = flush;
236 	}
237 
238 	/**
239 	 * If the provided value is greater than {@code 0}, then {@link javax.persistence.EntityManager#flush()}
240 	 * will be called after persistence operations as well as within batch operations.
241 	 * This property has precedence over the {@link #flush}, if it is specified to a value greater than {@code 0}.
242 	 * If the entity to persist is not an instance of {@link Iterable} and this property is greater than {@code 0},
243 	 * then the entity will be flushed as if the {@link #flush} attribute was set to {@code true}.
244 	 * @param flushSize defaults to '0'.
245 	 */
246 	public void setFlushSize(int flushSize) {
247 		Assert.state(flushSize >= 0, "'flushSize' cannot be less than '0'.");
248 		this.flushSize = flushSize;
249 	}
250 
251 	/**
252 	 * If set to {@code true} the {@link javax.persistence.EntityManager#clear()} will be called,
253 	 * and only if the {@link javax.persistence.EntityManager#flush()} was called after performing persistence
254 	 * operations.
255 	 * @param clearOnFlush defaults to 'false'.
256 	 * @see #setFlush(boolean)
257 	 * @see #setFlushSize(int)
258 	 */
259 	public void setClearOnFlush(boolean clearOnFlush) {
260 		this.clearOnFlush = clearOnFlush;
261 	}
262 
263 	/**
264 	 * If not set, this property defaults to <code>false</code>, which means that
265 	 * deletion occurs on a per object basis if a collection of entities is being
266 	 * deleted.
267 	 *<p>If set to 'true' the elements of the payload are deleted as a batch
268 	 * operation. Be aware that this exhibits issues in regards to cascaded deletes.
269 	 *<p>The specification 'JSR 317: Java Persistence API, Version 2.0' does not
270 	 * support cascaded deletes in batch operations. The specification states in
271 	 * chapter 4.10:
272 	 *<p>"A delete operation only applies to entities of the specified class and
273 	 * its subclasses. It does not cascade to related entities."
274 	 * @param deleteInBatch Defaults to 'false' if not set.
275 	 */
276 	public void setDeleteInBatch(boolean deleteInBatch) {
277 		this.deleteInBatch = deleteInBatch;
278 	}
279 
280 	/**
281 	 * If set to 'true', the retrieved objects are deleted from the database upon
282 	 * being polled. May not work in all situations, e.g. for Native SQL Queries.
283 	 * @param deleteAfterPoll Defaults to 'false'.
284 	 */
285 	public void setDeleteAfterPoll(boolean deleteAfterPoll) {
286 		this.deleteAfterPoll = deleteAfterPoll;
287 	}
288 
289 	/**
290 	 * @param parameterSourceFactory Must not be null
291 	 */
292 	public void setParameterSourceFactory(ParameterSourceFactory parameterSourceFactory) {
293 		Assert.notNull(parameterSourceFactory, "parameterSourceFactory must not be null.");
294 		this.parameterSourceFactory = parameterSourceFactory;
295 	}
296 
297 	/**
298 	 * Specify the {@link ParameterSource} that would be used to provide
299 	 * additional parameters.
300 	 * @param parameterSource Must not be null.
301 	 */
302 	public void setParameterSource(ParameterSource parameterSource) {
303 		Assert.notNull(parameterSource, "parameterSource must not be null.");
304 		this.parameterSource = parameterSource;
305 	}
306 
307 	/**
308 	 *
309 	 * This parameter indicates that only one result object shall be returned as
310 	 * a result from the executed JPA operation. If set to <code>true</code> and
311 	 * the result list from the JPA operations contains only 1 element, then that
312 	 * 1 element is extracted and returned as payload.
313 	 * <p>If the result map contains more than 1 element and
314 	 * {@link JpaExecutor#expectSingleResult} is <code>true</code>, then a
315 	 * {@link MessagingException} is thrown.
316 	 * <p>If set to <code>false</code>, the complete result list is returned as the
317 	 * payload.
318 	 * @param expectSingleResult true if a single object is expected.
319 	 *
320 	 */
321 	public void setExpectSingleResult(boolean expectSingleResult) {
322 		this.expectSingleResult = expectSingleResult;
323 	}
324 
325 	/**
326 	 * Set the expression that will be evaluated to get the first result in the query executed.
327 	 * If a null expression is set, all the results in the result set will be retrieved
328 	 * @param firstResultExpression The first result expression.
329 	 * @see javax.persistence.Query#setFirstResult(int)
330 	 */
331 	public void setFirstResultExpression(Expression firstResultExpression) {
332 		this.firstResultExpression = firstResultExpression;
333 	}
334 
335 	/**
336 	 * Set the expression that will be evaluated to get the {@code primaryKey} for
337 	 * {@link javax.persistence.EntityManager#find(Class, Object)}
338 	 * @param idExpression the SpEL expression for entity {@code primaryKey}.
339 	 * @since 4.0
340 	 */
341 	public void setIdExpression(Expression idExpression) {
342 		this.idExpression = idExpression;
343 	}
344 
345 	/**
346 	 * Set the expression for maximum number of results expression. It has be a non null value
347 	 * Not setting one will default to the behavior of fetching all the records
348 	 * @param maxResultsExpression The maximum results expression.
349 	 */
350 	public void setMaxResultsExpression(Expression maxResultsExpression) {
351 		Assert.notNull(maxResultsExpression, "maxResultsExpression cannot be null");
352 		this.maxResultsExpression = maxResultsExpression;
353 	}
354 
355 	/**
356 	 * Set the max number of results to retrieve from the database. Defaults to
357 	 * 0, which means that all possible objects shall be retrieved.
358 	 * @param maxNumberOfResults Must not be negative.
359 	 * @see javax.persistence.Query#setMaxResults(int)
360 	 */
361 	public void setMaxNumberOfResults(int maxNumberOfResults) {
362 		this.setMaxResultsExpression(new LiteralExpression("" + maxNumberOfResults));
363 	}
364 
365 	@Override
366 	public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
367 		this.beanFactory = beanFactory;
368 	}
369 
370 	/**
371 	 * Verify and sets the parameters. E.g. initializes the to be used
372 	 * {@link ParameterSourceFactory}.
373 	 */
374 	@Override
375 	public void afterPropertiesSet() {
376 		if (!CollectionUtils.isEmpty(this.jpaParameters)) {
377 			if (this.parameterSourceFactory == null) {
378 				ExpressionEvaluatingParameterSourceFactory expressionSourceFactory =
379 						new ExpressionEvaluatingParameterSourceFactory(this.beanFactory);
380 				expressionSourceFactory.setParameters(this.jpaParameters);
381 				this.parameterSourceFactory = expressionSourceFactory;
382 
383 			}
384 			else {
385 				throw new IllegalStateException("The 'jpaParameters' and 'parameterSourceFactory' " +
386 						"are mutually exclusive. Consider to configure parameters on the provided " +
387 						"'parameterSourceFactory': " + this.parameterSourceFactory);
388 			}
389 
390 			if (this.usePayloadAsParameterSource == null) {
391 				this.usePayloadAsParameterSource = false;
392 			}
393 
394 		}
395 		else {
396 
397 			if (this.parameterSourceFactory == null) {
398 				this.parameterSourceFactory = new BeanPropertyParameterSourceFactory();
399 			}
400 
401 			if (this.usePayloadAsParameterSource == null) {
402 				this.usePayloadAsParameterSource = true;
403 			}
404 		}
405 
406 		if (this.flushSize > 0) {
407 			this.flush = true;
408 		}
409 		else if (this.flush) {
410 			this.flushSize = 1;
411 		}
412 
413 		if (this.evaluationContext == null) {
414 			this.evaluationContext = ExpressionUtils.createStandardEvaluationContext(this.beanFactory);
415 		}
416 	}
417 
418 	/**
419 	 * Execute the actual Jpa Operation. Call this method, if you need access to
420 	 * process return values. This methods return a Map that contains either
421 	 * the number of affected entities or the affected entity itself.
422 	 *<p>Keep in mind that the number of entities effected by the operation may
423 	 * not necessarily correlate with the number of rows effected in the database.
424 	 * @param message The message.
425 	 * @return Either the number of affected entities when using a JPQL query.
426 	 * When using a merge/persist the updated/inserted itself is returned.
427 	 */
428 	public Object executeOutboundJpaOperation(Message<?> message) {
429 		ParameterSource paramSource = null;
430 		if (this.jpaQuery != null || this.nativeQuery != null || this.namedQuery != null) {
431 			paramSource = determineParameterSource(message);
432 		}
433 		if (this.jpaQuery != null) {
434 			return this.jpaOperations.executeUpdate(this.jpaQuery, paramSource);
435 		}
436 		else if (this.nativeQuery != null) {
437 			return this.jpaOperations.executeUpdateWithNativeQuery(this.nativeQuery, paramSource);
438 		}
439 		else if (this.namedQuery != null) {
440 			return this.jpaOperations.executeUpdateWithNamedQuery(this.namedQuery, paramSource);
441 		}
442 		else {
443 			return executeOutboundJpaOperationOnPersistentMode(message);
444 		}
445 	}
446 
447 	private Object executeOutboundJpaOperationOnPersistentMode(Message<?> message) {
448 		Object payload = message.getPayload();
449 		switch (this.persistMode) {
450 			case PERSIST:
451 				this.jpaOperations.persist(payload, this.flushSize, this.clearOnFlush);
452 				return payload;
453 			case MERGE:
454 				return this.jpaOperations.merge(payload, this.flushSize, this.clearOnFlush); // NOSONAR
455 			case DELETE:
456 				this.jpaOperations.delete(payload);
457 				if (this.flush) {
458 					this.jpaOperations.flush();
459 				}
460 				return payload;
461 			default:
462 				throw new IllegalStateException("Unsupported PersistMode: " + this.persistMode.name());
463 		}
464 	}
465 
466 	/**
467 	 * Execute the JPA operation. Delegates to {@link JpaExecutor#poll(Message)}.
468 	 * @return The object or null.
469 	 */
470 	@Nullable
471 	public Object poll() {
472 		return poll(null);
473 	}
474 
475 	/**
476 	 * Execute a (typically retrieving) JPA operation. The <i>requestMessage</i>
477 	 * can be used to provide additional query parameters using
478 	 * {@link JpaExecutor#parameterSourceFactory}. If the
479 	 * <i>requestMessage</i> parameter is null then
480 	 * {@link JpaExecutor#parameterSource} is being used for providing query parameters.
481 	 * @param requestMessage May be null.
482 	 * @return The payload object, which may be null.
483 	 */
484 	@Nullable
485 	public Object poll(@Nullable final Message<?> requestMessage) {
486 		final Object payload;
487 
488 		if (this.idExpression != null) {
489 			Object id = this.idExpression.getValue(this.evaluationContext, requestMessage); // NOSONAR It can be null
490 			Assert.state(id != null, "The 'idExpression' cannot evaluate to null.");
491 			Class<?> entityClazz = this.entityClass;
492 			if (entityClazz == null && requestMessage != null) {
493 				entityClazz = requestMessage.getPayload().getClass();
494 			}
495 			Assert.state(entityClazz != null, "The entity class to retrieve cannot be null.");
496 			payload = this.jpaOperations.find(entityClazz, id);
497 		}
498 		else {
499 			final List<?> result;
500 			int maxNumberOfResults = evaluateExpressionForNumericResult(requestMessage, this.maxResultsExpression);
501 			if (requestMessage == null) {
502 				result = doPoll(this.parameterSource, 0, maxNumberOfResults);
503 			}
504 			else {
505 				int firstResult = 0;
506 				if (this.firstResultExpression != null) {
507 					firstResult = getFirstResult(requestMessage);
508 				}
509 				ParameterSource paramSource = determineParameterSource(requestMessage);
510 				result = doPoll(paramSource, firstResult, maxNumberOfResults);
511 			}
512 
513 			if (result.isEmpty()) {
514 				payload = null;
515 			}
516 			else {
517 				if (this.expectSingleResult) {
518 					if (result.size() == 1) {
519 						payload = result.iterator().next();
520 					}
521 					else if (requestMessage != null) {
522 						throw new MessagingException(requestMessage,
523 								"The Jpa operation returned more than 1 result for expectSingleResult mode.");
524 					}
525 					else {
526 						throw new MessagingException(
527 								"The Jpa operation returned more than 1 result for expectSingleResult mode.");
528 					}
529 				}
530 				else {
531 					payload = result;
532 				}
533 			}
534 		}
535 
536 		checkDelete(payload);
537 		return payload;
538 	}
539 
540 	private void checkDelete(final Object payload) {
541 		if (payload != null && this.deleteAfterPoll) {
542 			if (payload instanceof Iterable) {
543 				if (this.deleteInBatch) {
544 					this.jpaOperations.deleteInBatch((Iterable<?>) payload);
545 				}
546 				else {
547 					for (Object entity : (Iterable<?>) payload) {
548 						this.jpaOperations.delete(entity);
549 					}
550 				}
551 			}
552 			else {
553 				this.jpaOperations.delete(payload);
554 			}
555 
556 			if (this.flush) {
557 				this.jpaOperations.flush();
558 			}
559 		}
560 	}
561 
562 	protected List<?> doPoll(ParameterSource jpaQLParameterSource, int firstResult, int maxNumberOfResults) {
563 		List<?> payload;
564 		if (this.jpaQuery != null) {
565 			payload =
566 					this.jpaOperations.getResultListForQuery(this.jpaQuery, jpaQLParameterSource,
567 							firstResult, maxNumberOfResults);
568 		}
569 		else if (this.nativeQuery != null) {
570 			payload =
571 					this.jpaOperations.getResultListForNativeQuery(this.nativeQuery, this.entityClass,
572 							jpaQLParameterSource, firstResult, maxNumberOfResults);
573 		}
574 		else if (this.namedQuery != null) {
575 			payload =
576 					this.jpaOperations.getResultListForNamedQuery(this.namedQuery, jpaQLParameterSource,
577 							firstResult, maxNumberOfResults);
578 		}
579 		else if (this.entityClass != null) {
580 			payload = this.jpaOperations.getResultListForClass(this.entityClass, firstResult, maxNumberOfResults);
581 		}
582 		else {
583 			throw new IllegalStateException("For the polling operation, one of "
584 					+ "the following properties must be specified: "
585 					+ "query, namedQuery or entityClass.");
586 		}
587 		return payload;
588 	}
589 
590 	private int getFirstResult(final Message<?> requestMessage) {
591 		return evaluateExpressionForNumericResult(requestMessage, this.firstResultExpression);
592 	}
593 
594 	private int evaluateExpressionForNumericResult(@Nullable final Message<?> requestMessage,
595 			@Nullable Expression expression) {
596 
597 		int evaluatedResult = 0;
598 		if (expression != null) {
599 			Object evaluationResult = expression.getValue(this.evaluationContext, requestMessage); // NOSONAR can be
600 			// null
601 			if (evaluationResult != null) {
602 				if (evaluationResult instanceof Number) {
603 					evaluatedResult = ((Number) evaluationResult).intValue();
604 				}
605 				else if (evaluationResult instanceof String) {
606 					try {
607 						evaluatedResult = Integer.parseInt((String) evaluationResult);
608 					}
609 					catch (NumberFormatException e) {
610 						throw new IllegalArgumentException(
611 								"Value " + evaluationResult + " passed as cannot be " +
612 										"parsed to a number, expected to be numeric", e);
613 					}
614 				}
615 				else {
616 					throw new IllegalArgumentException("Expected the value to be a Number got "
617 							+ evaluationResult.getClass().getName());
618 				}
619 			}
620 		}
621 		return evaluatedResult;
622 	}
623 
624 	private ParameterSource determineParameterSource(final Message<?> requestMessage) {
625 		if (this.usePayloadAsParameterSource) {
626 			return this.parameterSourceFactory.createParameterSource(requestMessage.getPayload());
627 		}
628 		else {
629 			return this.parameterSourceFactory.createParameterSource(requestMessage);
630 		}
631 	}
632 
633 }